**MAIN PROJECT**

**ROUTER 1x3 IN SYSTEM VERILOG**

**SUBMITTED BY**

**MELVIN RIJOHN T**

**03/02/2025**

TESTBENCH FILES 2

# TESTBENCH FILES

TRANSACTION

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: transaction.sv                         \*/

/\*      DESCRIPTION: Formatting data packet               \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

// Packet type enumeration for defining packet states

typedef enum logic[1:0]{RESET = 0, HEADER = 1, PAYLOAD = 2, PARITY = 3} pkt\_type\_t;

// Packet class definition

class Packet;

    // Randomizable fields representing packet attributes

    rand bit[7:0] header;

    rand bit[7:0] data;

    rand bit pkt\_valid;

    randc bit rd\_en\_0;

    randc bit rd\_en\_1;

    randc bit rd\_en\_2;

    // Status signals

    bit vld\_out\_0;

    bit vld\_out\_1;

    bit vld\_out\_2;

    bit err;

    bit busy;

    bit rst;

    // Data output signals

    bit [7:0] dout\_0;

    bit [7:0] dout\_1;

    bit [7:0] dout\_2;

    // Parity check logic

    logic[7:0] parity;

    // Packet type enumeration variable

    pkt\_type\_t pkt\_type;

    // Constraints to ensure valid header values

    constraint con1 {

        header[1:0] != 2'b11;

        header[7:2] != 0;

        header[7:2] <= 20;

    }

    // Function to copy data from another Packet instance

    function void copy(Packet tmp);

        data = tmp.data;

        pkt\_valid = tmp.pkt\_valid;

        rd\_en\_0 = tmp.rd\_en\_0;

        rd\_en\_1 = tmp.rd\_en\_1;

        rd\_en\_2 = tmp.rd\_en\_2;

        vld\_out\_0 = tmp.vld\_out\_0;

        vld\_out\_1 = tmp.vld\_out\_1;

        vld\_out\_2 = tmp.vld\_out\_2;

        err = tmp.err;

        busy = tmp.busy;

        dout\_0 = tmp.dout\_0;

        dout\_1 = tmp.dout\_1;

        dout\_2 = tmp.dout\_2;

        parity = tmp.parity;

    endfunction

endclass

GENERATOR

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: generator.sv                           \*/

/\*      DESCRIPTION: Generates input streams             \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Generator;

    mailbox #(Packet) mbx; // Mailbox for communication with driver

    event drv\_done; // Event to synchronize with driver

    Packet pkt; // Packet instance

    bit[7:0] header; // Header field storage

    function new(mailbox #(Packet) mbx, event drv\_done);

        this.mbx = mbx; // Initialize mailbox

        this.drv\_done = drv\_done; // Initialize event

    endfunction

    task run(int loopCount = 1);

        repeat(loopCount) begin

            pkt = new(); // Create a new packet instance

            pkt.parity = 0; // Initialize parity bit

            $display("[%0tps] Generator: Starting....", $time);

            pkt.pkt\_type = RESET; // Send reset packet

            mbx.put(pkt);

            @(drv\_done); // Wait for driver to complete

            if(!pkt.randomize()) $error("Randomization failed"); // Randomize packet fields

            pkt.pkt\_type = HEADER; // Set packet type to HEADER

            header = pkt.header; // Store header value

            pkt.parity = pkt.parity ^ pkt.header; // Compute parity

            mbx.put(pkt); // Send header packet

            @(drv\_done);

            for (int i = 0; i < header[7:2]; i++) begin // Loop through payload data

                if(!pkt.randomize()) $error("Randomization failed"); // Randomize payload data

                pkt.parity = pkt.parity ^ pkt.data; // Update parity

                pkt.pkt\_type = PAYLOAD; // Set packet type to PAYLOAD

                mbx.put(pkt); // Send payload packet

                @(drv\_done);

            end

            pkt.pkt\_type = PARITY; // Set packet type to PARITY

            pkt.data = pkt.parity; // Store computed parity in data field

            mbx.put(pkt); // Send parity packet

        end

    endtask

endclass

DRIVER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: driver.sv                             \*/

/\*      DESCRIPTION: Drives the input streams to the dut \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Driver;

    local bit[7:0] header; // Stores packet header data

    mailbox #(Packet) mbx; // Mailbox for communication with other components

    event drv\_done; // Event to signal when driving is done

    virtual router\_if vif; // Virtual interface for DUT interaction

    function new(mailbox #(Packet) mbx, event drv\_done, virtual router\_if vif);

        this.mbx = mbx; // Initialize mailbox

        this.drv\_done = drv\_done; // Initialize event

        this.vif = vif; // Initialize virtual interface

    endfunction

    task run();

        $display("[%0tps] Driver: Starting...", $time);

        forever begin

            Packet pkt = new(); // Create new packet instance

            mbx.get(pkt); // Retrieve packet from mailbox

            drive(pkt); // Drive packet data to DUT

            vif.rd\_en\_0 = pkt.rd\_en\_0; // Set read enable signals

            vif.rd\_en\_1 = pkt.rd\_en\_1;

            vif.rd\_en\_2 = pkt.rd\_en\_2;

            ->drv\_done; // Signal driver completion

        end

    endtask

    task drive(Packet pkt);

        case (pkt.pkt\_type)

            RESET: reset\_dut(); // Handle reset packet

            HEADER: drive\_header(pkt); // Handle header packet

            PAYLOAD: drive\_payload(pkt); // Handle payload packet

            PARITY: drive\_parity(pkt); // Handle parity packet

            default: $display("Invalid packet type"); // Handle invalid packet types

        endcase

    endtask

    task reset\_dut();

        vif.rst = 0; // Deassert reset

        @(posedge vif.clk);

        vif.rst = 1; // Assert reset

        vif.pkt\_valid = 0; // Deassert packet valid

        @(posedge vif.clk);

    endtask

    task drive\_header(Packet pkt);

        wait(vif.busy == 0); // Wait until DUT is not busy

        @(negedge vif.clk);

        vif.pkt\_valid = 1; // Assert packet valid

        vif.data = pkt.header; // Drive header data

        @(posedge vif.clk);

        @(posedge vif.clk);

    endtask

    task drive\_payload(Packet pkt);

        wait(vif.busy == 0); // Wait until DUT is not busy

        @(negedge vif.clk);

        vif.pkt\_valid <= 1; // Assert packet valid

        vif.data <= pkt.data; // Drive payload data

    endtask

    task drive\_parity(Packet pkt);

        wait(vif.busy == 0); // Wait until DUT is not busy

        @(negedge vif.clk);

        vif.pkt\_valid <= 0; // Deassert packet valid

        vif.data <= pkt.parity; // Drive parity data

    endtask

endclass

INTERFACE

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: interface.sv                           \*/

/\*      DESCRIPTION: Actual interface definition         \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

interface router\_if();

    logic clk;           // Clock signal

    logic rst;           // Reset signal

    logic [7:0] data;    // Data input

    logic pkt\_valid;     // Packet valid signal

    logic rd\_en\_0;       // Read enable signal for output 0

    logic rd\_en\_1;       // Read enable signal for output 1

    logic rd\_en\_2;       // Read enable signal for output 2

    logic vld\_out\_0;     // Valid output signal for output 0

    logic vld\_out\_1;     // Valid output signal for output 1

    logic vld\_out\_2;     // Valid output signal for output 2

    logic err;           // Error signal

    logic busy;          // Busy signal indicating router activity

    logic [7:0] dout\_0;  // Data output for output 0

    logic [7:0] dout\_1;  // Data output for output 1

    logic [7:0] dout\_2;  // Data output for output 2

    initial clk = 0; // Initialize clock to 0

    always #5 clk = ~clk; // Clock toggles every 5 time units

endinterface

MONITOR

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: monitor.sv                             \*/

/\*      DESCRIPTION: Mointors the output from the dut     \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Monitor;

    local bit[7:0] header = 0; // Stores the header value

    mailbox #(Packet) mbx\_in; // Mailbox for storing incoming packets

    mailbox #(Packet) mbx\_out; // Mailbox for storing outgoing packets

    event drv\_done; // Event to synchronize with the driver

    virtual router\_if vif; // Virtual interface for communication with DUT

    int count = 0; // Counter for incoming packets

    int prev\_val = 0; // Stores previous output value to detect changes

    function new(mailbox #(Packet) mbx\_in, mailbox #(Packet) mbx\_out, event drv\_done, virtual router\_if vif);

        this.mbx\_in = mbx\_in; // Initialize input mailbox

        this.mbx\_out = mbx\_out; // Initialize output mailbox

        this.drv\_done = drv\_done; // Initialize event

        this.vif = vif; // Initialize virtual interface

    endfunction

    task run();

        $display("[%0tps] Monitor: Starting...", $time);

        forever begin

            checkPacket\_in(header); // Check incoming packets

        end

    endtask

    task run1();

        checkPacket\_out(); // Check outgoing packets

    endtask

    task checkPacket\_in(ref bit[7:0] header);

        Packet item = new(); // Create a new packet instance

        @(drv\_done); // Wait for driver completion

        @(posedge vif.clk);

        #1;

        item = parsePacket(); // Parse packet data from interface

        if(!header && item.pkt\_valid) begin

            item.pkt\_type = HEADER; // Identify header packet

            header = item.data;

        end

        else if(header && item.pkt\_valid) begin

            item.pkt\_type = PAYLOAD; // Identify payload packet

        end

        else if(header && !item.pkt\_valid) begin

            item.pkt\_type = PARITY; // Identify parity packet

        end

        else begin

            item.pkt\_type = RESET; // Identify reset packet

        end

        mbx\_in.put(item); // Store packet in input mailbox

        count = count + 1; // Increment packet count

    endtask

    task checkPacket\_out();

        int count\_1 = 0;

        Packet item = new(); // Create a new packet instance

        while(count\_1 < header[7:2] + 1) begin // Process expected number of packets

            @(posedge vif.clk);

            #1;

            wait(vif.rd\_en\_0 || vif.rd\_en\_1 || vif.rd\_en\_2); // Wait for read enable signals

            item = parsePacket(); // Parse packet data from interface

            if(item.rd\_en\_0 && (item.dout\_0 != 0) && (prev\_val != item.dout\_0)) begin

                mbx\_out.put(item); // Store packet in output mailbox

                count\_1 = count\_1 + 1; // Increment processed packet count

                prev\_val = item.dout\_0; // Update previous value

            end

            else if(item.rd\_en\_1 && (item.dout\_1 != 0) && (prev\_val != item.dout\_1)) begin

                mbx\_out.put(item);

                count\_1 = count\_1 + 1;

                prev\_val = item.dout\_1;

            end

            else if (item.rd\_en\_2 && (item.dout\_2 != 0) && (prev\_val != item.dout\_2)) begin

                mbx\_out.put(item);

                count\_1 = count\_1 + 1;

                prev\_val = item.dout\_2;

            end else begin

                count\_1 = count\_1; // Maintain count if no new data

            end

        end

    endtask

    function Packet parsePacket();

        Packet pkt = new(); // Create a new packet instance

        pkt.rst = vif.rst;

        pkt.pkt\_valid = vif.pkt\_valid;

        pkt.data = vif.data;

        pkt.rd\_en\_0 = vif.rd\_en\_0;

        pkt.rd\_en\_1 = vif.rd\_en\_1;

        pkt.rd\_en\_2 = vif.rd\_en\_2;

        pkt.vld\_out\_0 = vif.vld\_out\_0;

        pkt.vld\_out\_1 = vif.vld\_out\_1;

        pkt.vld\_out\_2 = vif.vld\_out\_2;

        pkt.err = vif.err;

        pkt.busy = vif.busy;

        pkt.dout\_0 = vif.dout\_0;

        pkt.dout\_1 = vif.dout\_1;

        pkt.dout\_2 = vif.dout\_2;

        return pkt; // Return parsed packet

    endfunction

endclass

SCOREBOARD

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*   AUTHOR: METECH                                               \*/

/\*   FILE\_NAME: scoreboard.sv                                       \*/

/\*   DESCRIPTION: Verifies design using the received output and golden reference \*/

/\*   DATE: 03/02/2025                                               \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

class Scoreboard;

    bit[7:0] header = 0; // Stores the packet header

    mailbox #(Packet) mbx\_in; // Mailbox for input packets

    mailbox #(Packet) mbx\_out; // Mailbox for output packets

    virtual router\_if vif; // Virtual interface for router

    bit[7:0] in\_stream[$], out\_stream[$]; // Queues to store input and output data streams

    logic TX\_done = 0; // Transmission done flag

    logic RX\_done = 0; // Reception done flag

    int prev\_val = 0; // Stores previous value to prevent duplicates

    // Constructor: Initializes mailboxes

    function new(mailbox #(Packet) mbx\_in, mailbox #(Packet) mbx\_out);

        this.mbx\_in = mbx\_in;

        this.mbx\_out = mbx\_out;

    endfunction

    // Task to process incoming packets

    task in\_run();

        $display("[%0tps] Scoreboard: Starting...", $time);

        forever begin

            Packet pkt;

            if(mbx\_in.num() > 0) begin

                mbx\_in.get(pkt);

                checkPacket(pkt, this.header);

            end else begin

                #10; // Wait to avoid busy-waiting

            end

        end

    endtask

    // Task to process outgoing packets

    task out\_run();

        int count = 1;

        forever begin

            Packet pkt;

            if (mbx\_out.num() > 0) begin

                mbx\_out.get(pkt);

                checkPacket\_1(pkt);

                checkall();

                if (!(count > header[7:2] + 1)) begin

                   count = count + 1;

                end

            end else begin

                #10; // Prevent busy-waiting

            end

        end

    endtask

    // Task to check incoming packets and store them

    task checkPacket(Packet item, ref bit[7:0] header);

        bit[8:0] cnt;

        if(item.pkt\_valid && item.rst) begin

            if(item.pkt\_type == HEADER) begin

                header = item.data;

                cnt = header[7:2] + 2;

                in\_stream.push\_back(header);

            end

            if(item.pkt\_type == PAYLOAD || item.pkt\_type == PARITY) begin

                in\_stream.push\_back(item.data);

            end

        end

    endtask

    // Task to check outgoing packets and store them

    task checkPacket\_1(Packet item);

        if(item.rd\_en\_0 && (item.dout\_0 != 0) && (prev\_val != item.dout\_0)) begin

            out\_stream.push\_back(item.dout\_0);

            prev\_val = item.dout\_0;

        end

        else if(item.rd\_en\_1 && (item.dout\_1 != 0) && (prev\_val != item.dout\_1)) begin

            out\_stream.push\_back(item.dout\_1);

            prev\_val = item.dout\_1;

        end

        else if(item.rd\_en\_2 && (item.dout\_2 != 0) && (prev\_val != item.dout\_2)) begin

            out\_stream.push\_back(item.dout\_2);

            prev\_val = item.dout\_2;

        end

    endtask

    // Task to compare input and output streams for verification

    task checkall();

        int in\_parity = 0;

        int out\_parity = 0;

        if((in\_stream.size() == header[7:2] + 1) && (out\_stream.size() == header[7:2] + 1)) begin

            foreach(in\_stream[i]) begin

                in\_parity = in\_parity ^ in\_stream[i];

            end

            in\_stream.push\_back(in\_parity);

            foreach(out\_stream[i]) begin

                out\_parity = out\_parity ^ out\_stream[i];

            end

            out\_stream.push\_back(out\_parity);

            if(in\_parity == out\_parity) begin

                $display("/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/");

                $display("/\* Successfully verified Router 1x3");

                $display("/\* Input: %0p", in\_stream);

                $display("/\* Output: %0p", out\_stream);

                $display("/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/");

            end

            else begin

                $display("Verification unsuccessful: in\_parity = %0h, out\_parity = %0h", in\_parity, out\_parity);

            end

        end

    endtask

endclass

ENVIRONMENT

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                               \*/

/\*      FILE\_NAME: environment.sv                                     \*/

/\*      DESCRIPTION: Connects driver, generator, monitor &scoreboard \*/

/\*      DATE: 03/02/2025                                             \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

`include "transaction.sv"

`include "generator.sv"

`include "driver.sv"

`include "monitor.sv"

`include "scoreboard.sv"

// Environment class to instantiate and connect all verification components

class Environment;

    Generator gen; // Generates test packets

    Driver drv; // Drives packets to DUT

    Monitor mon; // Monitors DUT output

    Scoreboard sbd; // Compares expected vs actual results

    mailbox #(Packet) drv\_mbx; // Mailbox for driver communication

    mailbox #(Packet) sbd\_mbx\_in; // Mailbox for input packets to scoreboard

    mailbox #(Packet) sbd\_mbx\_out; // Mailbox for output packets from scoreboard

    event drv\_done; // Event to synchronize driver completion

    virtual router\_if vif; // Virtual interface for DUT interaction

    function new(virtual router\_if vif);

        drv\_mbx = new(); // Initialize driver mailbox

        sbd\_mbx\_in = new(); // Initialize input mailbox for scoreboard

        sbd\_mbx\_out = new(); // Initialize output mailbox for scoreboard

        gen = new(drv\_mbx, drv\_done); // Instantiate generator

        drv = new(drv\_mbx, drv\_done, vif); // Instantiate driver

        mon = new(sbd\_mbx\_in, sbd\_mbx\_out, drv\_done, vif); // Instantiate monitor

        sbd = new(sbd\_mbx\_in, sbd\_mbx\_out); // Instantiate scoreboard

    endfunction

    task run();

        fork

            gen.run(); // Run generator

            drv.run(); // Run driver

            mon.run(); // Run monitor

            mon.run1(); // Additional monitor function

            sbd.in\_run(); // Run input side of scoreboard

            sbd.out\_run(); // Run output side of scoreboard

        join

    endtask

endclass

TESTBENCH TOP

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: testbench.sv                         \*/

/\*      DESCRIPTION: Testbench top module                 \*/

/\*      DATE: 03/02/2025                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

`include "interface.sv"

`include "environment.sv"

module tb();

    router\_if intf(); // Instantiate the router interface

    Environment env = new(intf); // Create environment instance with the interface

    router dut(intf.clk, intf.rst, intf.data, intf.pkt\_valid, intf.rd\_en\_0, intf.rd\_en\_1, intf.rd\_en\_2, intf.vld\_out\_0, intf.vld\_out\_1, intf.vld\_out\_2, intf.err, intf.busy, intf.dout\_0, intf.dout\_1, intf.dout\_2); // Instantiate the router DUT

    initial begin

        env.run(); // Execute testbench environment

    end

    initial begin

        $dumpfile("out.vcd"); // Specify the VCD file for waveform dumping

        $dumpvars(1); // Dump all variables for debugging

        #2000 $finish; // Terminate simulation after 2000 time units

        end

endmodule

DESIGN (ROUTER 1x3)

ROUTER TOP

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: router.sv                             \*/

/\*      DESCRIPTION: Top level module of a 1x3 router   \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module router (

    input logic clk,                    // Clock input

    input logic rst,                    // Reset input

    input logic [7:0] d\_in,             // Data input (8 bits)

    input logic pkt\_valid,              // Packet validity signal

    input logic rd\_en\_0,                // Read enable for FIFO 0

    input logic rd\_en\_1,                // Read enable for FIFO 1

    input logic rd\_en\_2,                // Read enable for FIFO 2

    output logic vld\_out\_0,             // Valid output for FIFO 0

    output logic vld\_out\_1,             // Valid output for FIFO 1

    output logic vld\_out\_2,             // Valid output for FIFO 2

    output logic err,                   // Error signal

    output logic busy,                  // Busy signal indicating processing

    output logic [7:0] dout\_0,          // Data output from FIFO 0

    output logic [7:0] dout\_1,          // Data output from FIFO 1

    output logic [7:0] dout\_2           // Data output from FIFO 2

);

    // Internal wire declarations for FIFO control signals and state management

    logic soft\_rst\_0, full\_0, empty\_0;

    logic soft\_rst\_1, full\_1, empty\_1;

    logic soft\_rst\_2, full\_2, empty\_2;

    logic fifo\_full, detect\_addr, ld\_state, laf\_state;

    logic full\_state, lfd\_state, rst\_int\_reg;

    logic parity\_done, low\_pkt\_valid, wr\_en\_reg;

    logic [2:0] wr\_en;               // Write enable for the FIFOs

    logic [7:0] din;                 // Data input to FIFOs

    // Instantiate FIFOs

    fifo FIFO\_0 (clk, rst, soft\_rst\_0, wr\_en[0], rd\_en\_0, lfd\_state, din, full\_0, empty\_0, dout\_0);

    fifo FIFO\_1 (clk, rst, soft\_rst\_1, wr\_en[1], rd\_en\_1, lfd\_state, din, full\_1, empty\_1, dout\_1);

    fifo FIFO\_2 (clk, rst, soft\_rst\_2, wr\_en[2], rd\_en\_2, lfd\_state, din, full\_2, empty\_2, dout\_2);

    // Instantiate synchronizer to manage input data and FIFO states

    synchronizer SYNC (

        clk, rst, d\_in[1:0], detect\_addr,

        full\_0, full\_1, full\_2,

        empty\_0, empty\_1, empty\_2,

        wr\_en\_reg, rd\_en\_0, rd\_en\_1, rd\_en\_2,

        wr\_en, fifo\_full,

        vld\_out\_0, vld\_out\_1, vld\_out\_2,

        soft\_rst\_0, soft\_rst\_1, soft\_rst\_2

    );

    // Instantiate registers to store data and manage errors

    register REG\_0 (

        clk, rst, pkt\_valid, d\_in,

        fifo\_full, detect\_addr,

        ld\_state, laf\_state, full\_state, lfd\_state,

        rst\_int\_reg, din, err,

        parity\_done, low\_pkt\_valid

    );

    // Instantiate FSM controller to manage router states and operations

    fsm\_controller FSM (

        clk, rst, pkt\_valid, fifo\_full,

        empty\_0, empty\_1, empty\_2,

        soft\_rst\_0, soft\_rst\_1, soft\_rst\_2,

        parity\_done, low\_pkt\_valid,

        d\_in[1:0], wr\_en\_reg,

        detect\_addr, ld\_state, laf\_state,

        lfd\_state, full\_state,

        rst\_int\_reg, busy

    );

endmodule

FSM\_CONTROLLER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: fsm\_controller.sv                     \*/

/\*      DESCRIPTION:  FSM Controller module             \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module fsm\_controller (

    input logic clk,              // Clock input

    input logic rst,              // Active-low reset signal

    input logic pkt\_valid,        // Signal indicating a valid packet is present

    input logic fifo\_full,        // Signal indicating FIFO is full

    input logic fifo\_empty\_0,     // Signal indicating FIFO 0 is empty

    input logic fifo\_empty\_1,     // Signal indicating FIFO 1 is empty

    input logic fifo\_empty\_2,     // Signal indicating FIFO 2 is empty

    input logic soft\_rst\_0,       // Soft reset signal for FIFO 0

    input logic soft\_rst\_1,       // Soft reset signal for FIFO 1

    input logic soft\_rst\_2,       // Soft reset signal for FIFO 2

    input logic parity\_done,      // Signal indicating parity check is complete

    input logic low\_pkt\_valid,    // Signal indicating a low-valid packet condition

    input logic[1:0] din,        // 2-bit input specifying the destination FIFO

    output logic wr\_en\_req,       // Write enable request signal

    output logic detect\_addr,     // Signal to detect packet address

    output logic ld\_state,        // Load data state indicator

    output logic laf\_state,       // Load after full state indicator

    output logic lfd\_state,       // Load first data state indicator

    output logic full\_state,      // FIFO full state indicator

    output logic rst\_int\_reg,     // Reset internal register signal

    output logic busy             // Busy signal indicating FSM activity

);

  // State encoding for the FSM (1x3 router control)

  parameter DECODE\_ADDRESS     = 3'b000; // State to decode packet address

  parameter LOAD\_FIRST\_DATA    = 3'b001; // State to load the first data word

  parameter LOAD\_DATA          = 3'b010; // State to load subsequent data words

  parameter WAIT\_TILL\_EMPTY    = 3'b011; // Wait for the target FIFO to become empty

  parameter CHECK\_PARITY\_ERROR = 3'b100; // State to check for parity errors

  parameter LOAD\_PARITY        = 3'b101; // Load parity word

  parameter FIFO\_FULL\_STATE    = 3'b110; // State when FIFO is full

  parameter LOAD\_AFTER\_FULL    = 3'b111; // Load data after the FIFO becomes non-full

  logic [2:0] PS, NS; // Current State (PS) and Next State (NS) registers

  // State transition logic triggered on the rising edge of the clock

  always @(posedge clk) begin

    if (!rst)

      PS <= DECODE\_ADDRESS; // Reset state to DECODE\_ADDRESS

    else if (soft\_rst\_0 || soft\_rst\_1 || soft\_rst\_2)

      PS <= DECODE\_ADDRESS; // On any soft reset, transition to DECODE\_ADDRESS

    else

      PS <= NS; // Transition to the next state

  end

  // Next state logic based on the current state and input conditions

  always @(\*) begin

    NS = DECODE\_ADDRESS; // Default next state

    case (PS)

      DECODE\_ADDRESS: begin

        if ((pkt\_valid && din == 0 && fifo\_empty\_0) ||

            (pkt\_valid && din == 1 && fifo\_empty\_1) ||

            (pkt\_valid && din == 2 && fifo\_empty\_2))

          NS = LOAD\_FIRST\_DATA; // Load first data if FIFO is empty

        else if ((pkt\_valid && din == 0 && ~fifo\_empty\_0) ||

                 (pkt\_valid && din == 1 && !fifo\_empty\_1) ||

                 (pkt\_valid && din == 2 && !fifo\_empty\_2))

          NS = WAIT\_TILL\_EMPTY; // Wait if target FIFO is not empty

        else

          NS = DECODE\_ADDRESS; // Stay in the current state

      end

      LOAD\_FIRST\_DATA: NS = LOAD\_DATA; // Transition to LOAD\_DATA state

      LOAD\_DATA: begin

        if (fifo\_full)

          NS = FIFO\_FULL\_STATE; // If FIFO is full, transition to full state

        else if (!fifo\_full && !pkt\_valid)

          NS = LOAD\_PARITY; // If no more data, load parity

        else

          NS = LOAD\_DATA; // Continue loading data

      end

      WAIT\_TILL\_EMPTY: begin

        if (fifo\_empty\_0 || fifo\_empty\_1 || fifo\_empty\_2)

          NS = LOAD\_FIRST\_DATA; // If any FIFO becomes empty, load first data

        else

          NS = WAIT\_TILL\_EMPTY; // Continue waiting

      end

      FIFO\_FULL\_STATE: begin

        if (!fifo\_full)

          NS = LOAD\_AFTER\_FULL; // If FIFO is no longer full, load after full

        else

          NS = FIFO\_FULL\_STATE; // Stay in the full state

      end

      LOAD\_AFTER\_FULL: begin

        if (!parity\_done && !low\_pkt\_valid)

          NS = LOAD\_DATA; // If parity not done and valid, load data

        else if (!parity\_done && low\_pkt\_valid)

          NS = LOAD\_PARITY; // If low packet valid, load parity

        else if (parity\_done)

          NS = DECODE\_ADDRESS; // If parity done, decode next address

      end

      LOAD\_PARITY: NS = CHECK\_PARITY\_ERROR; // Transition to parity check

      CHECK\_PARITY\_ERROR: begin

        if (fifo\_full)

          NS = FIFO\_FULL\_STATE; // If FIFO is full, go to full state

        else

          NS = DECODE\_ADDRESS; // Otherwise, decode next address

      end

    endcase

  end

  // Output assignments based on the current state

  assign detect\_addr = (PS == DECODE\_ADDRESS); // Detect address in decode state

  assign wr\_en\_req = (PS == LOAD\_DATA || PS == LOAD\_PARITY || PS == LOAD\_AFTER\_FULL); // Write enable in specific states

  assign full\_state = (PS == FIFO\_FULL\_STATE); // Indicate FIFO full state

  assign lfd\_state = (PS == LOAD\_FIRST\_DATA); // Indicate load first data state

  assign busy = (PS == LOAD\_FIRST\_DATA || PS == LOAD\_PARITY || PS == FIFO\_FULL\_STATE || PS == LOAD\_AFTER\_FULL || PS == WAIT\_TILL\_EMPTY || PS == CHECK\_PARITY\_ERROR); // Indicate FSM is busy

  assign ld\_state = (PS == LOAD\_DATA); // Indicate load data state

  assign laf\_state = (PS == LOAD\_AFTER\_FULL); // Indicate load after full state

  assign rst\_int\_reg = (PS == CHECK\_PARITY\_ERROR); // Reset internal register during parity check

endmodule

REGISTER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                  \*/

/\*      FILE\_NAME: register.sv                           \*/

/\*      DESCRIPTION:  register module                   \*/

/\*      DATE: 21/12/2024                                \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module register (

    input logic clk,                // Clock input

    input logic rst,                // Active-low reset signal

    input logic pkt\_valid,          // Packet valid signal indicating data validity

    input logic [7:0] din,          // 8-bit data input

    input logic fifo\_full,          // Signal indicating if the FIFO is full

    input logic detect\_addr,        // Signal for address detection

    input logic ld\_state,           // Signal indicating load state is active

    input logic laf\_state,          // Signal indicating load after full state

    input logic full\_state,         // Signal indicating the full state of the system

    input logic lfd\_state,          // Signal indicating load first data state

    input logic rst\_int\_reg,        // Signal to reset the internal register

    output logic [7:0] dout,    // 8-bit data output

    output logic err,           // Error signal output

    output logic parity\_done,   // Parity check completion flag

    output logic low\_pkt\_valid  // Signal indicating low packet validity

);

  // Internal registers to store data, parity, and intermediate values

  logic [7:0] header, int\_reg, int\_parity, ext\_parity;

  // PARITY DONE LOGIC: Controls when the parity check is marked as done

  always @(posedge clk) begin

    if (!rst)

      parity\_done <= 0;       // Reset parity done flag

    else if (detect\_addr)

      parity\_done <= 0;       // Reset if address detection occurs

    else if ((ld\_state && (~fifo\_full) && (~pkt\_valid)) ||

             (laf\_state && low\_pkt\_valid && (~parity\_done)))

      parity\_done <= 1;       // Set parity done if conditions are met

  end

  // LOW PACKET VALID LOGIC: Manages the `low\_pkt\_valid` flag

  always @(posedge clk) begin

    if (!rst)

      low\_pkt\_valid <= 0;     // Reset low packet valid flag

    else if (rst\_int\_reg)

      low\_pkt\_valid <= 0;     // Reset if internal register is reset

    else if (ld\_state && ~pkt\_valid)

      low\_pkt\_valid <= 1;     // Set if in load state and no valid packet

  end

  // DATA OUT LOGIC: Controls the data output based on various states

  always @(posedge clk) begin

    if (!rst) begin

      dout <= 0;              // Reset data output

      header <= 0;            // Reset header register

      int\_reg <= 0;           // Reset internal register

    end else if (detect\_addr && pkt\_valid && din[1:0] != 2'b11)

      header <= din;          // Capture header if address is detected and packet is valid

    else if (lfd\_state)

      dout <= header;         // Output header if in load first data state

    else if (ld\_state && ~fifo\_full)

      dout <= din;            // Output data if in load state and FIFO is not full

    else if (ld\_state && fifo\_full)

      int\_reg <= din;         // Store data in internal register if FIFO is full

    else if (laf\_state)

      dout <= int\_reg;        // Output internal register data if in load after full state

  end

  // PARITY CALCULATE LOGIC: Computes the internal parity for error checking

  always @(posedge clk) begin

    if (!rst)

      int\_parity <= 0;      // Reset internal parity

    else if (detect\_addr)

      int\_parity <= 0;      // Reset if address detection occurs

    else if (lfd\_state && pkt\_valid)

      int\_parity <= int\_parity ^ header; // XOR with header data if packet is valid

    else if (ld\_state && pkt\_valid && ~full\_state)

      int\_parity <= int\_parity ^ din; // XOR with data input if in load state

    else

      int\_parity <= int\_parity; // Hold current parity value

  end

  // ERROR LOGIC: Checks if there is a parity error

  always @(posedge clk) begin

    if (!rst)

      err <= 0; // Reset error flag

    else if (parity\_done) begin

      if (int\_parity == ext\_parity)

        err <= 0; // No error if internal and external parity match

      else

        err <= 1; // Set error if parities do not match

    end else

      err <= 0; // Hold error as 0 if parity is not done

  end

  // EXTERNAL PARITY LOGIC: Stores the external parity value

  always @(posedge clk) begin

    if (!rst)

      ext\_parity <= 0; // Reset external parity

    else if (detect\_addr)

      ext\_parity <= 0; // Reset if address detection occurs

    else if ((ld\_state && !fifo\_full && !pkt\_valid) ||

             (laf\_state && ~parity\_done && low\_pkt\_valid))

      ext\_parity <= din; // Store data input as external parity if conditions are met

  end

endmodule

FIFO

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: fifo.sv                               \*/

/\*      DESCRIPTION:  16x9 Fifo Module                   \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module fifo (

    input logic clk,             // Clock input signal

    input logic rst,             // Active-low reset signal

    input logic soft\_reset,      // Soft reset signal to clear certain outputs

    input logic wr\_en,           // Write enable signal

    input logic rd\_en,           // Read enable signal

    input logic lfd\_state,       // State indicating the first data word (header)

    input logic [7:0] din,       // 8-bit input data to write into the FIFO

    output logic full,           // Full flag indicating the FIFO is full

    output logic empty,          // Empty flag indicating the FIFO is empty

    output logic [7:0] dout  // 8-bit output data from the FIFO

);

  logic [4:0] rd\_ptr, wr\_ptr;    // Read and write pointers (5 bits to track overflow)

  logic [6:0] intCount;          // Counter for tracking multi-byte packet size

  logic [8:0] mem[15:0];         // Memory array with 9-bit width for data + header bit                   //

  logic lfd\_state\_t;             // Temporary register to hold the lfd\_state signal

  // Latching lfd\_state signal with each clock cycle

  always @(posedge clk) begin

    if (!rst)

      lfd\_state\_t <= 0;        // Reset lfd\_state\_t when reset is active

    else

      lfd\_state\_t <= lfd\_state; // Store the current lfd\_state value

  end

  // Managing data output based on read enable and empty status

  always @(posedge clk) begin

    if (!rst)

      dout <= 8'b0;            // Reset dout to 0 on reset

    else if (soft\_reset)

      dout <= 8'bz;            // Set dout to high impedance on soft reset

    else if (rd\_en && !empty)

      dout <= mem[rd\_ptr[3:0]][7:0]; // Read data from memory if enabled and not empty

    else if (intCount == 0)

      dout <= 8'bz;            // High impedance when no data to output

  end

  // Memory write logic: Writing input data into the FIFO

  always @(posedge clk) begin

    if (!rst || soft\_reset) begin

      // Reset all memory locations on reset or soft reset

      for (int i = 0; i < 16; i = i + 1)

        mem[i] <= 0;

    end else if (wr\_en && !full) begin

      // Write data into memory if write enabled and not full

      if (lfd\_state\_t) begin

        mem[wr\_ptr[3:0]][8] <= 1'b1;  // Mark as header word

        mem[wr\_ptr[3:0]][7:0] <= din; // Store input data

      end else begin

        mem[wr\_ptr[3:0]][8] <= 1'b0;  // Mark as regular data word

        mem[wr\_ptr[3:0]][7:0] <= din; // Store input data

      end

    end

  end

  // Write pointer update logic

  always @(posedge clk) begin

    if (!rst)

      wr\_ptr <= 0;             // Reset write pointer

    else if (wr\_en && !full)

      wr\_ptr <= wr\_ptr + 1;    // Increment write pointer on write enable

  end

  // Read pointer update logic

  always @(posedge clk) begin

    if (!rst)

      rd\_ptr <= 0;             // Reset read pointer

    else if (rd\_en && !empty)

      rd\_ptr <= rd\_ptr + 1;    // Increment read pointer on read enable

  end

  // Internal counter management for tracking data packets

  always @(posedge clk) begin

    if (rd\_en && !empty) begin

      // If header word, initialize intCount with data size + 1

      if (mem[rd\_ptr[3:0]][8] == 1'b1)

        intCount <= mem[rd\_ptr[3:0]][7:2] + 1'b1;

      // Otherwise, decrement intCount if it's not zero

      else if (intCount != 0)

        intCount <= intCount - 1'b1;

    end

  end

  // Full flag: Set when write and read pointers overlap with different MSBs

  assign full = (wr\_ptr == {~rd\_ptr[4], rd\_ptr[3:0]});

  // Empty flag: Set when write and read pointers are identical

  assign empty = (rd\_ptr == wr\_ptr);

endmodule

SYNCHRONIZER

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

/\*      AUTHOR: METECH                                   \*/

/\*      FILE\_NAME: synchronizer.sv                       \*/

/\*      DESCRIPTION:  Synchronizer Module               \*/

/\*      DATE: 21/12/2024                                 \*/

/\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*/

module synchronizer (

    input logic clk,              // Clock input signal

    input logic rst,              // Active-low reset signal

    input logic[1:0] din,        // 2-bit input data to determine FIFO selection

    input logic detect\_addr,      // Signal indicating if address detection is active

    input logic full\_0,           // Signal indicating if FIFO 0 is full

    input logic full\_1,           // Signal indicating if FIFO 1 is full

    input logic full\_2,           // Signal indicating if FIFO 2 is full

    input logic empty\_0,          // Signal indicating if FIFO 0 is empty

    input logic empty\_1,          // Signal indicating if FIFO 1 is empty

    input logic empty\_2,          // Signal indicating if FIFO 2 is empty

    input logic wr\_en\_reg,        // Write enable register input

    input logic rd\_en\_0,          // Read enable signal for FIFO 0

    input logic rd\_en\_1,          // Read enable signal for FIFO 1

    input logic rd\_en\_2,          // Read enable signal for FIFO 2

    output logic [2:0] wr\_en, // 3-bit output to control write enable for each FIFO

    output logic fifo\_full,   // Output indicating if the selected FIFO is full

    output logic vld\_out\_0,       // Valid output signal for FIFO 0 (not empty)

    output logic vld\_out\_1,       // Valid output signal for FIFO 1 (not empty)

    output logic vld\_out\_2,       // Valid output signal for FIFO 2 (not empty)

    output logic soft\_reset\_0,// Soft reset signal for FIFO 0

    output logic soft\_reset\_1,// Soft reset signal for FIFO 1

    output logic soft\_reset\_2 // Soft reset signal for FIFO 2

);

  // Registers to count cycles for each FIFO's inactivity

  logic [5:0] count0, count1, count2;

  logic [1:0] tmp\_din; // Temporary register to hold the address input

  // Capture 'din' value on the detection of address

  always @(posedge clk) begin

    if (!rst)

      tmp\_din <= 0; // Reset 'tmp\_din' to 0 when reset is active

    else if (detect\_addr)

      tmp\_din <= din; // Store 'din' if address detection is active

  end

  // Control logic for write enable and FIFO full status based on 'tmp\_din'

  always @(\*) begin

    case (tmp\_din)

      2'b00: begin // Case for FIFO 0

        fifo\_full <= full\_0; // Set full status for FIFO 0

        wr\_en <= (wr\_en\_reg) ? 3'b001 : 0; // Enable write if 'wr\_en\_reg' is set

      end

      2'b01: begin // Case for FIFO 1

        fifo\_full <= full\_1; // Set full status for FIFO 1

        wr\_en <= (wr\_en\_reg) ? 3'b010 : 0; // Enable write if 'wr\_en\_reg' is set

      end

      2'b10: begin // Case for FIFO 2

        fifo\_full <= full\_2; // Set full status for FIFO 2

        wr\_en <= (wr\_en\_reg) ? 3'b100 : 0; // Enable write if 'wr\_en\_reg' is set

      end

      default: begin // Default case: no FIFO selected

        fifo\_full <= 0;

        wr\_en <= 0;

      end

    endcase

  end

  // Assign valid outputs based on FIFO emptiness

  assign vld\_out\_0 = (~empty\_0); // Valid if FIFO 0 is not empty

  assign vld\_out\_1 = (~empty\_1); // Valid if FIFO 1 is not empty

  assign vld\_out\_2 = (~empty\_2); // Valid if FIFO 2 is not empty

  // Monitor FIFO 0 for inactivity and trigger soft reset if needed

  always @(posedge clk) begin

    if (!rst) begin

      count0 <= 0;

      soft\_reset\_0 <= 0; // Reset state for FIFO 0

    end else if (vld\_out\_0) begin // If FIFO 0 has valid data

      if (!rd\_en\_0) begin // If not being read

        if (count0 == 29) begin // After 30 cycles

          soft\_reset\_0 <= 1; // Trigger soft reset

          count0 <= 0; // Reset counter

        end else begin

          soft\_reset\_0 <= 0;

          count0 <= count0 + 1; // Increment counter

        end

      end else

        count0 <= 0; // Reset counter if being read

    end

  end

  // Monitor FIFO 1 for inactivity and trigger soft reset if needed

  always @(posedge clk) begin

    if (!rst) begin

      count1 <= 0;

      soft\_reset\_1 <= 0; // Reset state for FIFO 1

    end else if (vld\_out\_1) begin // If FIFO 1 has valid data

      if (!rd\_en\_1) begin // If not being read

        if (count1 == 29) begin // After 30 cycles

          soft\_reset\_1 <= 1; // Trigger soft reset

          count1 <= 0; // Reset counter

        end else begin

          soft\_reset\_1 <= 0;

          count1 <= count1 + 1; // Increment counter

        end

      end else

        count1 <= 0; // Reset counter if being read

    end

  end

  // Monitor FIFO 2 for inactivity and trigger soft reset if needed

  always @(posedge clk) begin

    if (!rst) begin

      count2 <= 0;

      soft\_reset\_2 <= 0; // Reset state for FIFO 2

    end else if (vld\_out\_2) begin // If FIFO 2 has valid data

      if (!rd\_en\_2) begin // If not being read

        if (count2 == 29) begin // After 30 cycles

          soft\_reset\_2 <= 1; // Trigger soft reset

          count2 <= 0; // Reset counter

        end else begin

          soft\_reset\_2 <= 0;

          count2 <= count2 + 1; // Increment counter

        end

      end else

        count2 <= 0; // Reset counter if being read

    end

  end

endmodule

OUTPUT

![](data:image/png;base64,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)